**MAIN PROJECT**

**ROUTER 1x3 IN SYSTEM VERILOG**

**SUBMITTED BY**

**MELVIN RIJOHN T**

**03/02/2025**

TESTBENCH FILES 2

# TESTBENCH FILES

TRANSACTION

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: transaction.sv                         \*/

/\*      DESCRIPTION: Formatting data packet               \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

// Packet type enumeration for defining packet states

typedef enum logic[1:0]{RESET = 0, HEADER = 1, PAYLOAD = 2, PARITY = 3} pkt\_type\_t;

// Packet class definition

class Packet;

    // Randomizable fields representing packet attributes

    rand bit[7:0] header;

    rand bit[7:0] data;

    rand bit pkt\_valid;

    randc bit rd\_en\_0;

    randc bit rd\_en\_1;

    randc bit rd\_en\_2;

    // Status signals

    bit vld\_out\_0;

    bit vld\_out\_1;

    bit vld\_out\_2;

    bit err;

    bit busy;

    bit rst;

    // Data output signals

    bit [7:0] dout\_0;

    bit [7:0] dout\_1;

    bit [7:0] dout\_2;

    // Parity check logic

    logic[7:0] parity;

    // Packet type enumeration variable

    pkt\_type\_t pkt\_type;

    // Constraints to ensure valid header values

    constraint con1 {

        header[1:0] != 2'b11;

        header[7:2] != 0;

        header[7:2] <= 20;

    }

    // Function to copy data from another Packet instance

    function void copy(Packet tmp);

        data = tmp.data;

        pkt\_valid = tmp.pkt\_valid;

        rd\_en\_0 = tmp.rd\_en\_0;

        rd\_en\_1 = tmp.rd\_en\_1;

        rd\_en\_2 = tmp.rd\_en\_2;

        vld\_out\_0 = tmp.vld\_out\_0;

        vld\_out\_1 = tmp.vld\_out\_1;

        vld\_out\_2 = tmp.vld\_out\_2;

        err = tmp.err;

        busy = tmp.busy;

        dout\_0 = tmp.dout\_0;

        dout\_1 = tmp.dout\_1;

        dout\_2 = tmp.dout\_2;

        parity = tmp.parity;

    endfunction

endclass

GENERATOR

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: generator.sv                           \*/

/\*      DESCRIPTION: Generates input streams             \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Generator;

    mailbox #(Packet) mbx; // Mailbox for communication with driver

    event drv\_done; // Event to synchronize with driver

    Packet pkt; // Packet instance

    bit[7:0] header; // Header field storage

    function new(mailbox #(Packet) mbx, event drv\_done);

        this.mbx = mbx; // Initialize mailbox

        this.drv\_done = drv\_done; // Initialize event

    endfunction

    task run(int loopCount = 1);

        repeat(loopCount) begin

            pkt = new(); // Create a new packet instance

            pkt.parity = 0; // Initialize parity bit

            $display("[%0tps] Generator: Starting....", $time);

            pkt.pkt\_type = RESET; // Send reset packet

            mbx.put(pkt);

            @(drv\_done); // Wait for driver to complete

            if(!pkt.randomize()) $error("Randomization failed"); // Randomize packet fields

            pkt.pkt\_type = HEADER; // Set packet type to HEADER

            header = pkt.header; // Store header value

            pkt.parity = pkt.parity ^ pkt.header; // Compute parity

            mbx.put(pkt); // Send header packet

            @(drv\_done);

            for (int i = 0; i < header[7:2]; i++) begin // Loop through payload data

                if(!pkt.randomize()) $error("Randomization failed"); // Randomize payload data

                pkt.parity = pkt.parity ^ pkt.data; // Update parity

                pkt.pkt\_type = PAYLOAD; // Set packet type to PAYLOAD

                mbx.put(pkt); // Send payload packet

                @(drv\_done);

            end

            pkt.pkt\_type = PARITY; // Set packet type to PARITY

            pkt.data = pkt.parity; // Store computed parity in data field

            mbx.put(pkt); // Send parity packet

        end

    endtask

endclass

DRIVER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: driver.sv                             \*/

/\*      DESCRIPTION: Drives the input streams to the dut \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Driver;

    local bit[7:0] header; // Stores packet header data

    mailbox #(Packet) mbx; // Mailbox for communication with other components

    event drv\_done; // Event to signal when driving is done

    virtual router\_if vif; // Virtual interface for DUT interaction

    function new(mailbox #(Packet) mbx, event drv\_done, virtual router\_if vif);

        this.mbx = mbx; // Initialize mailbox

        this.drv\_done = drv\_done; // Initialize event

        this.vif = vif; // Initialize virtual interface

    endfunction

    task run();

        $display("[%0tps] Driver: Starting...", $time);

        forever begin

            Packet pkt = new(); // Create new packet instance

            mbx.get(pkt); // Retrieve packet from mailbox

            drive(pkt); // Drive packet data to DUT

            vif.rd\_en\_0 = pkt.rd\_en\_0; // Set read enable signals

            vif.rd\_en\_1 = pkt.rd\_en\_1;

            vif.rd\_en\_2 = pkt.rd\_en\_2;

            ->drv\_done; // Signal driver completion

        end

    endtask

    task drive(Packet pkt);

        case (pkt.pkt\_type)

            RESET: reset\_dut(); // Handle reset packet

            HEADER: drive\_header(pkt); // Handle header packet

            PAYLOAD: drive\_payload(pkt); // Handle payload packet

            PARITY: drive\_parity(pkt); // Handle parity packet

            default: $display("Invalid packet type"); // Handle invalid packet types

        endcase

    endtask

    task reset\_dut();

        vif.rst = 0; // Deassert reset

        @(posedge vif.clk);

        vif.rst = 1; // Assert reset

        vif.pkt\_valid = 0; // Deassert packet valid

        @(posedge vif.clk);

    endtask

    task drive\_header(Packet pkt);

        wait(vif.busy == 0); // Wait until DUT is not busy

        @(negedge vif.clk);

        vif.pkt\_valid = 1; // Assert packet valid

        vif.data = pkt.header; // Drive header data

        @(posedge vif.clk);

        @(posedge vif.clk);

    endtask

    task drive\_payload(Packet pkt);

        wait(vif.busy == 0); // Wait until DUT is not busy

        @(negedge vif.clk);

        vif.pkt\_valid <= 1; // Assert packet valid

        vif.data <= pkt.data; // Drive payload data

    endtask

    task drive\_parity(Packet pkt);

        wait(vif.busy == 0); // Wait until DUT is not busy

        @(negedge vif.clk);

        vif.pkt\_valid <= 0; // Deassert packet valid

        vif.data <= pkt.parity; // Drive parity data

    endtask

endclass

INTERFACE

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: interface.sv                           \*/

/\*      DESCRIPTION: Actual interface definition         \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

interface router\_if();

    logic clk;           // Clock signal

    logic rst;           // Reset signal

    logic [7:0] data;    // Data input

    logic pkt\_valid;     // Packet valid signal

    logic rd\_en\_0;       // Read enable signal for output 0

    logic rd\_en\_1;       // Read enable signal for output 1

    logic rd\_en\_2;       // Read enable signal for output 2

    logic vld\_out\_0;     // Valid output signal for output 0

    logic vld\_out\_1;     // Valid output signal for output 1

    logic vld\_out\_2;     // Valid output signal for output 2

    logic err;           // Error signal

    logic busy;          // Busy signal indicating router activity

    logic [7:0] dout\_0;  // Data output for output 0

    logic [7:0] dout\_1;  // Data output for output 1

    logic [7:0] dout\_2;  // Data output for output 2

    initial clk = 0; // Initialize clock to 0

    always #5 clk = ~clk; // Clock toggles every 5 time units

endinterface

MONITOR

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: monitor.sv                             \*/

/\*      DESCRIPTION: Mointors the output from the dut     \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Monitor;

    local bit[7:0] header = 0; // Stores the header value

    mailbox #(Packet) mbx\_in; // Mailbox for storing incoming packets

    mailbox #(Packet) mbx\_out; // Mailbox for storing outgoing packets

    event drv\_done; // Event to synchronize with the driver

    virtual router\_if vif; // Virtual interface for communication with DUT

    int count = 0; // Counter for incoming packets

    int prev\_val = 0; // Stores previous output value to detect changes

    function new(mailbox #(Packet) mbx\_in, mailbox #(Packet) mbx\_out, event drv\_done, virtual router\_if vif);

        this.mbx\_in = mbx\_in; // Initialize input mailbox

        this.mbx\_out = mbx\_out; // Initialize output mailbox

        this.drv\_done = drv\_done; // Initialize event

        this.vif = vif; // Initialize virtual interface

    endfunction

    task run();

        $display("[%0tps] Monitor: Starting...", $time);

        forever begin

            checkPacket\_in(header); // Check incoming packets

        end

    endtask

    task run1();

        checkPacket\_out(); // Check outgoing packets

    endtask

    task checkPacket\_in(ref bit[7:0] header);

        Packet item = new(); // Create a new packet instance

        @(drv\_done); // Wait for driver completion

        @(posedge vif.clk);

        #1;

        item = parsePacket(); // Parse packet data from interface

        if(!header && item.pkt\_valid) begin

            item.pkt\_type = HEADER; // Identify header packet

            header = item.data;

        end

        else if(header && item.pkt\_valid) begin

            item.pkt\_type = PAYLOAD; // Identify payload packet

        end

        else if(header && !item.pkt\_valid) begin

            item.pkt\_type = PARITY; // Identify parity packet

        end

        else begin

            item.pkt\_type = RESET; // Identify reset packet

        end

        mbx\_in.put(item); // Store packet in input mailbox

        count = count + 1; // Increment packet count

    endtask

    task checkPacket\_out();

        int count\_1 = 0;

        Packet item = new(); // Create a new packet instance

        while(count\_1 < header[7:2] + 1) begin // Process expected number of packets

            @(posedge vif.clk);

            #1;

            wait(vif.rd\_en\_0 || vif.rd\_en\_1 || vif.rd\_en\_2); // Wait for read enable signals

            item = parsePacket(); // Parse packet data from interface

            if(item.rd\_en\_0 && (item.dout\_0 != 0) && (prev\_val != item.dout\_0)) begin

                mbx\_out.put(item); // Store packet in output mailbox

                count\_1 = count\_1 + 1; // Increment processed packet count

                prev\_val = item.dout\_0; // Update previous value

            end

            else if(item.rd\_en\_1 && (item.dout\_1 != 0) && (prev\_val != item.dout\_1)) begin

                mbx\_out.put(item);

                count\_1 = count\_1 + 1;

                prev\_val = item.dout\_1;

            end

            else if (item.rd\_en\_2 && (item.dout\_2 != 0) && (prev\_val != item.dout\_2)) begin

                mbx\_out.put(item);

                count\_1 = count\_1 + 1;

                prev\_val = item.dout\_2;

            end else begin

                count\_1 = count\_1; // Maintain count if no new data

            end

        end

    endtask

    function Packet parsePacket();

        Packet pkt = new(); // Create a new packet instance

        pkt.rst = vif.rst;

        pkt.pkt\_valid = vif.pkt\_valid;

        pkt.data = vif.data;

        pkt.rd\_en\_0 = vif.rd\_en\_0;

        pkt.rd\_en\_1 = vif.rd\_en\_1;

        pkt.rd\_en\_2 = vif.rd\_en\_2;

        pkt.vld\_out\_0 = vif.vld\_out\_0;

        pkt.vld\_out\_1 = vif.vld\_out\_1;

        pkt.vld\_out\_2 = vif.vld\_out\_2;

        pkt.err = vif.err;

        pkt.busy = vif.busy;

        pkt.dout\_0 = vif.dout\_0;

        pkt.dout\_1 = vif.dout\_1;

        pkt.dout\_2 = vif.dout\_2;

        return pkt; // Return parsed packet

    endfunction

endclass

SCOREBOARD

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*   AUTHOR: METECH                                               \*/

/\*   FILE\_NAME: scoreboard.sv                                       \*/

/\*   DESCRIPTION: Verifies design using the received output and golden reference \*/

/\*   DATE: 03/02/2025                                               \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Scoreboard;

    bit[7:0] header = 0; // Stores the packet header

    mailbox #(Packet) mbx\_in; // Mailbox for input packets

    mailbox #(Packet) mbx\_out; // Mailbox for output packets

    virtual router\_if vif; // Virtual interface for router

    bit[7:0] in\_stream[$], out\_stream[$]; // Queues to store input and output data streams

    logic TX\_done = 0; // Transmission done flag

    logic RX\_done = 0; // Reception done flag

    int prev\_val = 0; // Stores previous value to prevent duplicates

    // Constructor: Initializes mailboxes

    function new(mailbox #(Packet) mbx\_in, mailbox #(Packet) mbx\_out);

        this.mbx\_in = mbx\_in;

        this.mbx\_out = mbx\_out;

    endfunction

    // Task to process incoming packets

    task in\_run();

        $display("[%0tps] Scoreboard: Starting...", $time);

        forever begin

            Packet pkt;

            if(mbx\_in.num() > 0) begin

                mbx\_in.get(pkt);

                checkPacket(pkt, this.header);

            end else begin

                #10; // Wait to avoid busy-waiting

            end

        end

    endtask

    // Task to process outgoing packets

    task out\_run();

        int count = 1;

        forever begin

            Packet pkt;

            if (mbx\_out.num() > 0) begin

                mbx\_out.get(pkt);

                checkPacket\_1(pkt);

                checkall();

                if (!(count > header[7:2] + 1)) begin

                   count = count + 1;

                end

            end else begin

                #10; // Prevent busy-waiting

            end

        end

    endtask

    // Task to check incoming packets and store them

    task checkPacket(Packet item, ref bit[7:0] header);

        bit[8:0] cnt;

        if(item.pkt\_valid && item.rst) begin

            if(item.pkt\_type == HEADER) begin

                header = item.data;

                cnt = header[7:2] + 2;

                in\_stream.push\_back(header);

            end

            if(item.pkt\_type == PAYLOAD || item.pkt\_type == PARITY) begin

                in\_stream.push\_back(item.data);

            end

        end

    endtask

    // Task to check outgoing packets and store them

    task checkPacket\_1(Packet item);

        if(item.rd\_en\_0 && (item.dout\_0 != 0) && (prev\_val != item.dout\_0)) begin

            out\_stream.push\_back(item.dout\_0);

            prev\_val = item.dout\_0;

        end

        else if(item.rd\_en\_1 && (item.dout\_1 != 0) && (prev\_val != item.dout\_1)) begin

            out\_stream.push\_back(item.dout\_1);

            prev\_val = item.dout\_1;

        end

        else if(item.rd\_en\_2 && (item.dout\_2 != 0) && (prev\_val != item.dout\_2)) begin

            out\_stream.push\_back(item.dout\_2);

            prev\_val = item.dout\_2;

        end

    endtask

    // Task to compare input and output streams for verification

    task checkall();

        int in\_parity = 0;

        int out\_parity = 0;

        if((in\_stream.size() == header[7:2] + 1) && (out\_stream.size() == header[7:2] + 1)) begin

            foreach(in\_stream[i]) begin

                in\_parity = in\_parity ^ in\_stream[i];

            end

            in\_stream.push\_back(in\_parity);

            foreach(out\_stream[i]) begin

                out\_parity = out\_parity ^ out\_stream[i];

            end

            out\_stream.push\_back(out\_parity);

            if(in\_parity == out\_parity) begin

                $display("/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/");

                $display("/\* Successfully verified Router 1x3");

                $display("/\* Input: %0p", in\_stream);

                $display("/\* Output: %0p", out\_stream);

                $display("/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/");

            end

            else begin

                $display("Verification unsuccessful: in\_parity = %0h, out\_parity = %0h", in\_parity, out\_parity);

            end

        end

    endtask

endclass

ENVIRONMENT

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                               \*/

/\*      FILE\_NAME: environment.sv                                     \*/

/\*      DESCRIPTION: Connects driver, generator, monitor &scoreboard \*/

/\*      DATE: 03/02/2025                                             \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

`include "transaction.sv"

`include "generator.sv"

`include "driver.sv"

`include "monitor.sv"

`include "scoreboard.sv"

// Environment class to instantiate and connect all verification components

class Environment;

    Generator gen; // Generates test packets

    Driver drv; // Drives packets to DUT

    Monitor mon; // Monitors DUT output

    Scoreboard sbd; // Compares expected vs actual results

    mailbox #(Packet) drv\_mbx; // Mailbox for driver communication

    mailbox #(Packet) sbd\_mbx\_in; // Mailbox for input packets to scoreboard

    mailbox #(Packet) sbd\_mbx\_out; // Mailbox for output packets from scoreboard

    event drv\_done; // Event to synchronize driver completion

    virtual router\_if vif; // Virtual interface for DUT interaction

    function new(virtual router\_if vif);

        drv\_mbx = new(); // Initialize driver mailbox

        sbd\_mbx\_in = new(); // Initialize input mailbox for scoreboard

        sbd\_mbx\_out = new(); // Initialize output mailbox for scoreboard

        gen = new(drv\_mbx, drv\_done); // Instantiate generator

        drv = new(drv\_mbx, drv\_done, vif); // Instantiate driver

        mon = new(sbd\_mbx\_in, sbd\_mbx\_out, drv\_done, vif); // Instantiate monitor

        sbd = new(sbd\_mbx\_in, sbd\_mbx\_out); // Instantiate scoreboard

    endfunction

    task run();

        fork

            gen.run(); // Run generator

            drv.run(); // Run driver

            mon.run(); // Run monitor

            mon.run1(); // Additional monitor function

            sbd.in\_run(); // Run input side of scoreboard

            sbd.out\_run(); // Run output side of scoreboard

        join

    endtask

endclass

TESTBENCH TOP

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: testbench.sv                         \*/

/\*      DESCRIPTION: Testbench top module                 \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

`include "interface.sv"

`include "environment.sv"

module tb();

    router\_if intf(); // Instantiate the router interface

    Environment env = new(intf); // Create environment instance with the interface

    router dut(intf.clk, intf.rst, intf.data, intf.pkt\_valid, intf.rd\_en\_0, intf.rd\_en\_1, intf.rd\_en\_2, intf.vld\_out\_0, intf.vld\_out\_1, intf.vld\_out\_2, intf.err, intf.busy, intf.dout\_0, intf.dout\_1, intf.dout\_2); // Instantiate the router DUT

    initial begin

        env.run(); // Execute testbench environment

    end

    initial begin

        $dumpfile("out.vcd"); // Specify the VCD file for waveform dumping

        $dumpvars(1); // Dump all variables for debugging

        #2000 $finish; // Terminate simulation after 2000 time units

        end

endmodule

DESIGN (ROUTER 1x3)

ROUTER TOP

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: router.sv                             \*/

/\*      DESCRIPTION: Top level module of a 1x3 router   \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module router (

    input logic clk,                    // Clock input

    input logic rst,                    // Reset input

    input logic [7:0] d\_in,             // Data input (8 bits)

    input logic pkt\_valid,              // Packet validity signal

    input logic rd\_en\_0,                // Read enable for FIFO 0

    input logic rd\_en\_1,                // Read enable for FIFO 1

    input logic rd\_en\_2,                // Read enable for FIFO 2

    output logic vld\_out\_0,             // Valid output for FIFO 0

    output logic vld\_out\_1,             // Valid output for FIFO 1

    output logic vld\_out\_2,             // Valid output for FIFO 2

    output logic err,                   // Error signal

    output logic busy,                  // Busy signal indicating processing

    output logic [7:0] dout\_0,          // Data output from FIFO 0

    output logic [7:0] dout\_1,          // Data output from FIFO 1

    output logic [7:0] dout\_2           // Data output from FIFO 2

);

    // Internal wire declarations for FIFO control signals and state management

    logic soft\_rst\_0, full\_0, empty\_0;

    logic soft\_rst\_1, full\_1, empty\_1;

    logic soft\_rst\_2, full\_2, empty\_2;

    logic fifo\_full, detect\_addr, ld\_state, laf\_state;

    logic full\_state, lfd\_state, rst\_int\_reg;

    logic parity\_done, low\_pkt\_valid, wr\_en\_reg;

    logic [2:0] wr\_en;               // Write enable for the FIFOs

    logic [7:0] din;                 // Data input to FIFOs

    // Instantiate FIFOs

    fifo FIFO\_0 (clk, rst, soft\_rst\_0, wr\_en[0], rd\_en\_0, lfd\_state, din, full\_0, empty\_0, dout\_0);

    fifo FIFO\_1 (clk, rst, soft\_rst\_1, wr\_en[1], rd\_en\_1, lfd\_state, din, full\_1, empty\_1, dout\_1);

    fifo FIFO\_2 (clk, rst, soft\_rst\_2, wr\_en[2], rd\_en\_2, lfd\_state, din, full\_2, empty\_2, dout\_2);

    // Instantiate synchronizer to manage input data and FIFO states

    synchronizer SYNC (

        clk, rst, d\_in[1:0], detect\_addr,

        full\_0, full\_1, full\_2,

        empty\_0, empty\_1, empty\_2,

        wr\_en\_reg, rd\_en\_0, rd\_en\_1, rd\_en\_2,

        wr\_en, fifo\_full,

        vld\_out\_0, vld\_out\_1, vld\_out\_2,

        soft\_rst\_0, soft\_rst\_1, soft\_rst\_2

    );

    // Instantiate registers to store data and manage errors

    register REG\_0 (

        clk, rst, pkt\_valid, d\_in,

        fifo\_full, detect\_addr,

        ld\_state, laf\_state, full\_state, lfd\_state,

        rst\_int\_reg, din, err,

        parity\_done, low\_pkt\_valid

    );

    // Instantiate FSM controller to manage router states and operations

    fsm\_controller FSM (

        clk, rst, pkt\_valid, fifo\_full,

        empty\_0, empty\_1, empty\_2,

        soft\_rst\_0, soft\_rst\_1, soft\_rst\_2,

        parity\_done, low\_pkt\_valid,

        d\_in[1:0], wr\_en\_reg,

        detect\_addr, ld\_state, laf\_state,

        lfd\_state, full\_state,

        rst\_int\_reg, busy

    );

endmodule

FSM\_CONTROLLER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: fsm\_controller.sv                     \*/

/\*      DESCRIPTION:  FSM Controller module             \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module fsm\_controller (

    input logic clk,              // Clock input

    input logic rst,              // Active-low reset signal

    input logic pkt\_valid,        // Signal indicating a valid packet is present

    input logic fifo\_full,        // Signal indicating FIFO is full

    input logic fifo\_empty\_0,     // Signal indicating FIFO 0 is empty

    input logic fifo\_empty\_1,     // Signal indicating FIFO 1 is empty

    input logic fifo\_empty\_2,     // Signal indicating FIFO 2 is empty

    input logic soft\_rst\_0,       // Soft reset signal for FIFO 0

    input logic soft\_rst\_1,       // Soft reset signal for FIFO 1

    input logic soft\_rst\_2,       // Soft reset signal for FIFO 2

    input logic parity\_done,      // Signal indicating parity check is complete

    input logic low\_pkt\_valid,    // Signal indicating a low-valid packet condition

    input logic[1:0] din,        // 2-bit input specifying the destination FIFO

    output logic wr\_en\_req,       // Write enable request signal

    output logic detect\_addr,     // Signal to detect packet address

    output logic ld\_state,        // Load data state indicator

    output logic laf\_state,       // Load after full state indicator

    output logic lfd\_state,       // Load first data state indicator

    output logic full\_state,      // FIFO full state indicator

    output logic rst\_int\_reg,     // Reset internal register signal

    output logic busy             // Busy signal indicating FSM activity

);

  // State encoding for the FSM (1x3 router control)

  parameter DECODE\_ADDRESS     = 3'b000; // State to decode packet address

  parameter LOAD\_FIRST\_DATA    = 3'b001; // State to load the first data word

  parameter LOAD\_DATA          = 3'b010; // State to load subsequent data words

  parameter WAIT\_TILL\_EMPTY    = 3'b011; // Wait for the target FIFO to become empty

  parameter CHECK\_PARITY\_ERROR = 3'b100; // State to check for parity errors

  parameter LOAD\_PARITY        = 3'b101; // Load parity word

  parameter FIFO\_FULL\_STATE    = 3'b110; // State when FIFO is full

  parameter LOAD\_AFTER\_FULL    = 3'b111; // Load data after the FIFO becomes non-full

  logic [2:0] PS, NS; // Current State (PS) and Next State (NS) registers

  // State transition logic triggered on the rising edge of the clock

  always @(posedge clk) begin

    if (!rst)

      PS <= DECODE\_ADDRESS; // Reset state to DECODE\_ADDRESS

    else if (soft\_rst\_0 || soft\_rst\_1 || soft\_rst\_2)

      PS <= DECODE\_ADDRESS; // On any soft reset, transition to DECODE\_ADDRESS

    else

      PS <= NS; // Transition to the next state

  end

  // Next state logic based on the current state and input conditions

  always @(\*) begin

    NS = DECODE\_ADDRESS; // Default next state

    case (PS)

      DECODE\_ADDRESS: begin

        if ((pkt\_valid && din == 0 && fifo\_empty\_0) ||

            (pkt\_valid && din == 1 && fifo\_empty\_1) ||

            (pkt\_valid && din == 2 && fifo\_empty\_2))

          NS = LOAD\_FIRST\_DATA; // Load first data if FIFO is empty

        else if ((pkt\_valid && din == 0 && ~fifo\_empty\_0) ||

                 (pkt\_valid && din == 1 && !fifo\_empty\_1) ||

                 (pkt\_valid && din == 2 && !fifo\_empty\_2))

          NS = WAIT\_TILL\_EMPTY; // Wait if target FIFO is not empty

        else

          NS = DECODE\_ADDRESS; // Stay in the current state

      end

      LOAD\_FIRST\_DATA: NS = LOAD\_DATA; // Transition to LOAD\_DATA state

      LOAD\_DATA: begin

        if (fifo\_full)

          NS = FIFO\_FULL\_STATE; // If FIFO is full, transition to full state

        else if (!fifo\_full && !pkt\_valid)

          NS = LOAD\_PARITY; // If no more data, load parity

        else

          NS = LOAD\_DATA; // Continue loading data

      end

      WAIT\_TILL\_EMPTY: begin

        if (fifo\_empty\_0 || fifo\_empty\_1 || fifo\_empty\_2)

          NS = LOAD\_FIRST\_DATA; // If any FIFO becomes empty, load first data

        else

          NS = WAIT\_TILL\_EMPTY; // Continue waiting

      end

      FIFO\_FULL\_STATE: begin

        if (!fifo\_full)

          NS = LOAD\_AFTER\_FULL; // If FIFO is no longer full, load after full

        else

          NS = FIFO\_FULL\_STATE; // Stay in the full state

      end

      LOAD\_AFTER\_FULL: begin

        if (!parity\_done && !low\_pkt\_valid)

          NS = LOAD\_DATA; // If parity not done and valid, load data

        else if (!parity\_done && low\_pkt\_valid)

          NS = LOAD\_PARITY; // If low packet valid, load parity

        else if (parity\_done)

          NS = DECODE\_ADDRESS; // If parity done, decode next address

      end

      LOAD\_PARITY: NS = CHECK\_PARITY\_ERROR; // Transition to parity check

      CHECK\_PARITY\_ERROR: begin

        if (fifo\_full)

          NS = FIFO\_FULL\_STATE; // If FIFO is full, go to full state

        else

          NS = DECODE\_ADDRESS; // Otherwise, decode next address

      end

    endcase

  end

  // Output assignments based on the current state

  assign detect\_addr = (PS == DECODE\_ADDRESS); // Detect address in decode state

  assign wr\_en\_req = (PS == LOAD\_DATA || PS == LOAD\_PARITY || PS == LOAD\_AFTER\_FULL); // Write enable in specific states

  assign full\_state = (PS == FIFO\_FULL\_STATE); // Indicate FIFO full state

  assign lfd\_state = (PS == LOAD\_FIRST\_DATA); // Indicate load first data state

  assign busy = (PS == LOAD\_FIRST\_DATA || PS == LOAD\_PARITY || PS == FIFO\_FULL\_STATE || PS == LOAD\_AFTER\_FULL || PS == WAIT\_TILL\_EMPTY || PS == CHECK\_PARITY\_ERROR); // Indicate FSM is busy

  assign ld\_state = (PS == LOAD\_DATA); // Indicate load data state

  assign laf\_state = (PS == LOAD\_AFTER\_FULL); // Indicate load after full state

  assign rst\_int\_reg = (PS == CHECK\_PARITY\_ERROR); // Reset internal register during parity check

endmodule

REGISTER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                  \*/

/\*      FILE\_NAME: register.sv                           \*/

/\*      DESCRIPTION:  register module                   \*/

/\*      DATE: 21/12/2024                                \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module register (

    input logic clk,                // Clock input

    input logic rst,                // Active-low reset signal

    input logic pkt\_valid,          // Packet valid signal indicating data validity

    input logic [7:0] din,          // 8-bit data input

    input logic fifo\_full,          // Signal indicating if the FIFO is full

    input logic detect\_addr,        // Signal for address detection

    input logic ld\_state,           // Signal indicating load state is active

    input logic laf\_state,          // Signal indicating load after full state

    input logic full\_state,         // Signal indicating the full state of the system

    input logic lfd\_state,          // Signal indicating load first data state

    input logic rst\_int\_reg,        // Signal to reset the internal register

    output logic [7:0] dout,    // 8-bit data output

    output logic err,           // Error signal output

    output logic parity\_done,   // Parity check completion flag

    output logic low\_pkt\_valid  // Signal indicating low packet validity

);

  // Internal registers to store data, parity, and intermediate values

  logic [7:0] header, int\_reg, int\_parity, ext\_parity;

  // PARITY DONE LOGIC: Controls when the parity check is marked as done

  always @(posedge clk) begin

    if (!rst)

      parity\_done <= 0;       // Reset parity done flag

    else if (detect\_addr)

      parity\_done <= 0;       // Reset if address detection occurs

    else if ((ld\_state && (~fifo\_full) && (~pkt\_valid)) ||

             (laf\_state && low\_pkt\_valid && (~parity\_done)))

      parity\_done <= 1;       // Set parity done if conditions are met

  end

  // LOW PACKET VALID LOGIC: Manages the `low\_pkt\_valid` flag

  always @(posedge clk) begin

    if (!rst)

      low\_pkt\_valid <= 0;     // Reset low packet valid flag

    else if (rst\_int\_reg)

      low\_pkt\_valid <= 0;     // Reset if internal register is reset

    else if (ld\_state && ~pkt\_valid)

      low\_pkt\_valid <= 1;     // Set if in load state and no valid packet

  end

  // DATA OUT LOGIC: Controls the data output based on various states

  always @(posedge clk) begin

    if (!rst) begin

      dout <= 0;              // Reset data output

      header <= 0;            // Reset header register

      int\_reg <= 0;           // Reset internal register

    end else if (detect\_addr && pkt\_valid && din[1:0] != 2'b11)

      header <= din;          // Capture header if address is detected and packet is valid

    else if (lfd\_state)

      dout <= header;         // Output header if in load first data state

    else if (ld\_state && ~fifo\_full)

      dout <= din;            // Output data if in load state and FIFO is not full

    else if (ld\_state && fifo\_full)

      int\_reg <= din;         // Store data in internal register if FIFO is full

    else if (laf\_state)

      dout <= int\_reg;        // Output internal register data if in load after full state

  end

  // PARITY CALCULATE LOGIC: Computes the internal parity for error checking

  always @(posedge clk) begin

    if (!rst)

      int\_parity <= 0;      // Reset internal parity

    else if (detect\_addr)

      int\_parity <= 0;      // Reset if address detection occurs

    else if (lfd\_state && pkt\_valid)

      int\_parity <= int\_parity ^ header; // XOR with header data if packet is valid

    else if (ld\_state && pkt\_valid && ~full\_state)

      int\_parity <= int\_parity ^ din; // XOR with data input if in load state

    else

      int\_parity <= int\_parity; // Hold current parity value

  end

  // ERROR LOGIC: Checks if there is a parity error

  always @(posedge clk) begin

    if (!rst)

      err <= 0; // Reset error flag

    else if (parity\_done) begin

      if (int\_parity == ext\_parity)

        err <= 0; // No error if internal and external parity match

      else

        err <= 1; // Set error if parities do not match

    end else

      err <= 0; // Hold error as 0 if parity is not done

  end

  // EXTERNAL PARITY LOGIC: Stores the external parity value

  always @(posedge clk) begin

    if (!rst)

      ext\_parity <= 0; // Reset external parity

    else if (detect\_addr)

      ext\_parity <= 0; // Reset if address detection occurs

    else if ((ld\_state && !fifo\_full && !pkt\_valid) ||

             (laf\_state && ~parity\_done && low\_pkt\_valid))

      ext\_parity <= din; // Store data input as external parity if conditions are met

  end

endmodule

FIFO

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: fifo.sv                               \*/

/\*      DESCRIPTION:  16x9 Fifo Module                   \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module fifo (

    input logic clk,             // Clock input signal

    input logic rst,             // Active-low reset signal

    input logic soft\_reset,      // Soft reset signal to clear certain outputs

    input logic wr\_en,           // Write enable signal

    input logic rd\_en,           // Read enable signal

    input logic lfd\_state,       // State indicating the first data word (header)

    input logic [7:0] din,       // 8-bit input data to write into the FIFO

    output logic full,           // Full flag indicating the FIFO is full

    output logic empty,          // Empty flag indicating the FIFO is empty

    output logic [7:0] dout  // 8-bit output data from the FIFO

);

  logic [4:0] rd\_ptr, wr\_ptr;    // Read and write pointers (5 bits to track overflow)

  logic [6:0] intCount;          // Counter for tracking multi-byte packet size

  logic [8:0] mem[15:0];         // Memory array with 9-bit width for data + header bit                   //

  logic lfd\_state\_t;             // Temporary register to hold the lfd\_state signal

  // Latching lfd\_state signal with each clock cycle

  always @(posedge clk) begin

    if (!rst)

      lfd\_state\_t <= 0;        // Reset lfd\_state\_t when reset is active

    else

      lfd\_state\_t <= lfd\_state; // Store the current lfd\_state value

  end

  // Managing data output based on read enable and empty status

  always @(posedge clk) begin

    if (!rst)

      dout <= 8'b0;            // Reset dout to 0 on reset

    else if (soft\_reset)

      dout <= 8'bz;            // Set dout to high impedance on soft reset

    else if (rd\_en && !empty)

      dout <= mem[rd\_ptr[3:0]][7:0]; // Read data from memory if enabled and not empty

    else if (intCount == 0)

      dout <= 8'bz;            // High impedance when no data to output

  end

  // Memory write logic: Writing input data into the FIFO

  always @(posedge clk) begin

    if (!rst || soft\_reset) begin

      // Reset all memory locations on reset or soft reset

      for (int i = 0; i < 16; i = i + 1)

        mem[i] <= 0;

    end else if (wr\_en && !full) begin

      // Write data into memory if write enabled and not full

      if (lfd\_state\_t) begin

        mem[wr\_ptr[3:0]][8] <= 1'b1;  // Mark as header word

        mem[wr\_ptr[3:0]][7:0] <= din; // Store input data

      end else begin

        mem[wr\_ptr[3:0]][8] <= 1'b0;  // Mark as regular data word

        mem[wr\_ptr[3:0]][7:0] <= din; // Store input data

      end

    end

  end

  // Write pointer update logic

  always @(posedge clk) begin

    if (!rst)

      wr\_ptr <= 0;             // Reset write pointer

    else if (wr\_en && !full)

      wr\_ptr <= wr\_ptr + 1;    // Increment write pointer on write enable

  end

  // Read pointer update logic

  always @(posedge clk) begin

    if (!rst)

      rd\_ptr <= 0;             // Reset read pointer

    else if (rd\_en && !empty)

      rd\_ptr <= rd\_ptr + 1;    // Increment read pointer on read enable

  end

  // Internal counter management for tracking data packets

  always @(posedge clk) begin

    if (rd\_en && !empty) begin

      // If header word, initialize intCount with data size + 1

      if (mem[rd\_ptr[3:0]][8] == 1'b1)

        intCount <= mem[rd\_ptr[3:0]][7:2] + 1'b1;

      // Otherwise, decrement intCount if it's not zero

      else if (intCount != 0)

        intCount <= intCount - 1'b1;

    end

  end

  // Full flag: Set when write and read pointers overlap with different MSBs

  assign full = (wr\_ptr == {~rd\_ptr[4], rd\_ptr[3:0]});

  // Empty flag: Set when write and read pointers are identical

  assign empty = (rd\_ptr == wr\_ptr);

endmodule

SYNCHRONIZER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: synchronizer.sv                       \*/

/\*      DESCRIPTION:  Synchronizer Module               \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module synchronizer (

    input logic clk,              // Clock input signal

    input logic rst,              // Active-low reset signal

    input logic[1:0] din,        // 2-bit input data to determine FIFO selection

    input logic detect\_addr,      // Signal indicating if address detection is active

    input logic full\_0,           // Signal indicating if FIFO 0 is full

    input logic full\_1,           // Signal indicating if FIFO 1 is full

    input logic full\_2,           // Signal indicating if FIFO 2 is full

    input logic empty\_0,          // Signal indicating if FIFO 0 is empty

    input logic empty\_1,          // Signal indicating if FIFO 1 is empty

    input logic empty\_2,          // Signal indicating if FIFO 2 is empty

    input logic wr\_en\_reg,        // Write enable register input

    input logic rd\_en\_0,          // Read enable signal for FIFO 0

    input logic rd\_en\_1,          // Read enable signal for FIFO 1

    input logic rd\_en\_2,          // Read enable signal for FIFO 2

    output logic [2:0] wr\_en, // 3-bit output to control write enable for each FIFO

    output logic fifo\_full,   // Output indicating if the selected FIFO is full

    output logic vld\_out\_0,       // Valid output signal for FIFO 0 (not empty)

    output logic vld\_out\_1,       // Valid output signal for FIFO 1 (not empty)

    output logic vld\_out\_2,       // Valid output signal for FIFO 2 (not empty)

    output logic soft\_reset\_0,// Soft reset signal for FIFO 0

    output logic soft\_reset\_1,// Soft reset signal for FIFO 1

    output logic soft\_reset\_2 // Soft reset signal for FIFO 2

);

  // Registers to count cycles for each FIFO's inactivity

  logic [5:0] count0, count1, count2;

  logic [1:0] tmp\_din; // Temporary register to hold the address input

  // Capture 'din' value on the detection of address

  always @(posedge clk) begin

    if (!rst)

      tmp\_din <= 0; // Reset 'tmp\_din' to 0 when reset is active

    else if (detect\_addr)

      tmp\_din <= din; // Store 'din' if address detection is active

  end

  // Control logic for write enable and FIFO full status based on 'tmp\_din'

  always @(\*) begin

    case (tmp\_din)

      2'b00: begin // Case for FIFO 0

        fifo\_full <= full\_0; // Set full status for FIFO 0

        wr\_en <= (wr\_en\_reg) ? 3'b001 : 0; // Enable write if 'wr\_en\_reg' is set

      end

      2'b01: begin // Case for FIFO 1

        fifo\_full <= full\_1; // Set full status for FIFO 1

        wr\_en <= (wr\_en\_reg) ? 3'b010 : 0; // Enable write if 'wr\_en\_reg' is set

      end

      2'b10: begin // Case for FIFO 2

        fifo\_full <= full\_2; // Set full status for FIFO 2

        wr\_en <= (wr\_en\_reg) ? 3'b100 : 0; // Enable write if 'wr\_en\_reg' is set

      end

      default: begin // Default case: no FIFO selected

        fifo\_full <= 0;

        wr\_en <= 0;

      end

    endcase

  end

  // Assign valid outputs based on FIFO emptiness

  assign vld\_out\_0 = (~empty\_0); // Valid if FIFO 0 is not empty

  assign vld\_out\_1 = (~empty\_1); // Valid if FIFO 1 is not empty

  assign vld\_out\_2 = (~empty\_2); // Valid if FIFO 2 is not empty

  // Monitor FIFO 0 for inactivity and trigger soft reset if needed

  always @(posedge clk) begin

    if (!rst) begin

      count0 <= 0;

      soft\_reset\_0 <= 0; // Reset state for FIFO 0

    end else if (vld\_out\_0) begin // If FIFO 0 has valid data

      if (!rd\_en\_0) begin // If not being read

        if (count0 == 29) begin // After 30 cycles

          soft\_reset\_0 <= 1; // Trigger soft reset

          count0 <= 0; // Reset counter

        end else begin

          soft\_reset\_0 <= 0;

          count0 <= count0 + 1; // Increment counter

        end

      end else

        count0 <= 0; // Reset counter if being read

    end

  end

  // Monitor FIFO 1 for inactivity and trigger soft reset if needed

  always @(posedge clk) begin

    if (!rst) begin

      count1 <= 0;

      soft\_reset\_1 <= 0; // Reset state for FIFO 1

    end else if (vld\_out\_1) begin // If FIFO 1 has valid data

      if (!rd\_en\_1) begin // If not being read

        if (count1 == 29) begin // After 30 cycles

          soft\_reset\_1 <= 1; // Trigger soft reset

          count1 <= 0; // Reset counter

        end else begin

          soft\_reset\_1 <= 0;

          count1 <= count1 + 1; // Increment counter

        end

      end else

        count1 <= 0; // Reset counter if being read

    end

  end

  // Monitor FIFO 2 for inactivity and trigger soft reset if needed

  always @(posedge clk) begin

    if (!rst) begin

      count2 <= 0;

      soft\_reset\_2 <= 0; // Reset state for FIFO 2

    end else if (vld\_out\_2) begin // If FIFO 2 has valid data

      if (!rd\_en\_2) begin // If not being read

        if (count2 == 29) begin // After 30 cycles

          soft\_reset\_2 <= 1; // Trigger soft reset

          count2 <= 0; // Reset counter

        end else begin

          soft\_reset\_2 <= 0;

          count2 <= count2 + 1; // Increment counter

        end

      end else

        count2 <= 0; // Reset counter if being read

    end

  end

endmodule

OUTPUT

![](data:image/png;base64,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)